home *** CD-ROM | disk | FTP | other *** search
/ CU Amiga Super CD-ROM 11 / CU Amiga Magazine's Super CD-ROM 11 (1997)(EMAP Images)(GB)(Track 1 of 3)[!][issue 1997-06].iso / cucd / programming / oberonv4 / source / system / texts.mod (.txt) < prev    next >
Oberon Text  |  1996-08-11  |  31KB  |  802 lines

  1. Syntax10.Scn.Fnt
  2. Syntax10b.Scn.Fnt
  3. FoldElems
  4. ParcElems
  5. Alloc
  6. MODULE Texts;    (* Copyright (c) ETH Z
  7. rich, 1989-96 / jg, nw, cas, hm, rc 23.9.93 *)
  8.     See, and update if necessary the history at the bottom of the file.
  9. IMPORT
  10.     HostSYS
  11.     ,Display,Files,Fonts,Modules,Reals;
  12. (*--- insert field e: Elem into Texts.Scanner and change Texts.Scan to set it in case of class=6 *)
  13. CONST
  14.     ElemChar* = 1CX;
  15.     TAB = 9X; CR = 0DX; maxD = 9;
  16.     (**FileMsg.id**)
  17.         load* = 0; store* = 1;
  18.     (**Notifier op**)
  19.         replace* = 0; insert* = 1; delete* = 2;
  20.     (**Scanner.class**)
  21.         Inval* = 0; Name* = 1; String* = 2; Int* = 3; Real* = 4; LongReal* = 5; Char* = 6;
  22.     textTag=0F0X; textVersion=01X;
  23.         A run is either a piece or an elem.
  24.         The runs are enchained into a doubly linked list with a sentinel element,
  25.         to form the whole text. 
  26.         It is not clear, if the sentinel needs to have len=MAX(LONGINT), or if the
  27.         length is ignored.
  28.     Run = POINTER TO RunDesc;
  29.     RunDesc = RECORD
  30.         prev, next: Run;
  31.         len: LONGINT;    (* length of this run *)
  32.         fnt: Fonts.Font;
  33.         col, voff: SHORTINT;
  34.         host: BOOLEAN    (* remeber whether file was Oberon Text or host text *)
  35.     END;
  36.         The storage for a piece is always supplied by a file, either the file where
  37.         the Text came from, or a temporary file.
  38.     Piece = POINTER TO PieceDesc;
  39.     PieceDesc = RECORD (RunDesc)
  40.         file: Files.File;
  41.         org: LONGINT
  42.     END;
  43.     Elem* = POINTER TO ElemDesc;
  44.     Buffer* = POINTER TO BufDesc;
  45.     Text* = POINTER TO TextDesc;
  46.     ElemMsg* = RECORD END;
  47.     Handler* = PROCEDURE (e: Elem; VAR msg: ElemMsg);
  48.         base    points to the text which contains this element.
  49.     ElemDesc* = RECORD (RunDesc)
  50.         W*, H*: LONGINT;
  51.         handle*: Handler;
  52.         base: Text
  53.     END;
  54.     FileMsg* = RECORD (ElemMsg)
  55.         id*: INTEGER;
  56.         pos*: LONGINT;
  57.         r*: Files.Rider
  58.     END;
  59.     CopyMsg* = RECORD (ElemMsg)
  60.         e*: Elem
  61.     END;
  62.         Why aren't mod and proc types exported from Modules ???
  63.     IdentifyMsg* = RECORD (ElemMsg)
  64.         mod*, proc*: ARRAY 32 OF CHAR
  65.     END;
  66.     BufDesc* = RECORD
  67.         len*: LONGINT;
  68.         head: Run
  69.     END;
  70.     Notifier* = PROCEDURE (T: Text; op: INTEGER; beg, end: LONGINT);
  71.         head points to the sentinel.
  72.     TextDesc* = RECORD
  73.         len*: LONGINT;
  74.         notify*: Notifier;
  75.         head,cache: Run;
  76.         corg: LONGINT
  77.     END;
  78.     Reader* = RECORD
  79.         eot*: BOOLEAN;
  80.         fnt*: Fonts.Font;
  81.         col*, voff*: SHORTINT;
  82.         elem*: Elem;
  83.         rider: Files.Rider;
  84.         run: Run;
  85.         org, off: LONGINT
  86.     END;
  87.     Scanner* = RECORD (Reader)
  88.         nextCh*: CHAR;
  89.         line*, class*: INTEGER;
  90.         i*: LONGINT;
  91.         x*: REAL;
  92.         y*: LONGREAL;
  93.         c*: CHAR;
  94.         len*: INTEGER;
  95.         s*: ARRAY 512 OF CHAR;
  96.     END;
  97.     Writer* = RECORD
  98.         buf*: Buffer;
  99.         fnt*: Fonts.Font;
  100.         col*, voff*: SHORTINT;
  101.         rider: Files.Rider;
  102.         file: Files.File
  103.     END;
  104.         Why aren't mod and proc types exported from Modules ???
  105.     Alien = POINTER TO RECORD (ElemDesc)
  106.         file: Files.File;
  107.         org, span: LONGINT;
  108.         mod, proc: ARRAY 32 OF CHAR
  109.     END;
  110.     isNameChar-:PROCEDURE(ch:CHAR):BOOLEAN;
  111.     isNameChar1-:PROCEDURE(ch:CHAR):BOOLEAN;
  112.     new*: Elem;
  113.     del: Buffer;
  114. (* run primitives *)
  115. PROCEDURE Find (T: Text; VAR pos: LONGINT; VAR u: Run; VAR org, off: LONGINT); (*
  116.     Return the run which contains character/element at position pos.
  117.     off    Position within the run.
  118.     org    Position of start of run within Text.
  119. VAR v: Run; m: LONGINT;
  120. BEGIN
  121.     IF pos >= T.len THEN pos := T.len; u := T.head; org := T.len; off := 0; T.cache := T.head; T.corg := 0
  122.     ELSE v := T.cache.next; m := pos - T.corg;
  123.         IF pos >= T.corg THEN
  124.             WHILE m >= v.len DO DEC(m, v.len); v := v.next END
  125.         ELSE
  126.             WHILE m < 0 DO v := v.prev; INC(m, v.len) END;
  127.         END;
  128.         u := v; org := pos - m; off := m; T.cache := v.prev; T.corg := org
  129. END Find;
  130. PROCEDURE Split (off: LONGINT; VAR u, un: Run); (*
  131.     Split piece at relative posisiont off.
  132.     un    points to piece starting at the split point.
  133.     u    points to piece preceeding un.
  134.     p, U: Piece;
  135. BEGIN
  136.     IF off = 0 THEN
  137.         un := u; u := un.prev
  138.     ELSIF
  139.         off >= u.len THEN un := u.next
  140.     ELSE
  141.         NEW(p); un := p; U := u(Piece);
  142.         p^ := U^; INC(p.org, off); DEC(p.len, off); DEC(U.len, p.len);
  143.         p.host := u.host; p.prev := U; p.next := U.next; p.next.prev := p; U.next := p;
  144. END Split;
  145. PROCEDURE Merge (T:Text; u:Run; VAR v:Run); (*
  146.     Merge two pieces, if they are adjacent, use the same file for storage
  147.     and have the same attributes.
  148.     p,q:Piece;
  149. BEGIN
  150.     IF (u IS Piece) & (v IS Piece) & (u.fnt.name = v.fnt.name) & (u.col = v.col) & (u.voff = v.voff)
  151.     & (u(Piece).host = v(Piece).host) THEN
  152.         p := u(Piece); q := v(Piece);
  153.         IF (p.file = q.file) & (p.org + p.len = q.org) THEN
  154.             IF T.cache = u THEN INC(T.corg, q.len)
  155.             ELSIF T.cache = v THEN T.cache := T.head; T.corg := 0
  156.             END;
  157.             INC(p.len, q.len); v := v.next
  158.         END
  159. END Merge;
  160. PROCEDURE Splice (un, v, w: Run; base: Text);    (* (u, un) -> (u, v, w, un) *)
  161.     VAR u: Run;
  162. BEGIN
  163.     IF v # w.next THEN u := un.prev;
  164.         u.next := v; v.prev := u; un.prev := w; w.next := un;
  165.         REPEAT
  166.             IF v IS Elem THEN v(Elem).base := base END;
  167.             v := v.next
  168.         UNTIL v = un
  169. END Splice;
  170. PROCEDURE ClonePiece (p: Piece): Piece;
  171.     q:Piece;
  172. BEGIN
  173.     NEW(q);
  174.     q^ := p^;
  175.     RETURN q
  176. END ClonePiece;
  177. PROCEDURE CloneElem (e: Elem): Elem;
  178.     VAR msg: CopyMsg;
  179. BEGIN msg.e := NIL; e.handle(e, msg); RETURN msg.e
  180. END CloneElem;
  181. (** Elements **)
  182. PROCEDURE CopyElem* (SE, DE: Elem);
  183. BEGIN
  184.     DE.len := SE.len; DE.fnt := SE.fnt; DE.col := SE.col; DE.voff := SE.voff;
  185.     DE.W := SE.W; DE.H := SE.H; DE.handle := SE.handle
  186. END CopyElem;
  187. PROCEDURE ElemBase* (E: Elem): Text;
  188. BEGIN
  189.     RETURN E.base
  190. END ElemBase;
  191. PROCEDURE ElemPos* (E: Elem): LONGINT;
  192.     VAR u: Run; pos: LONGINT;
  193. BEGIN u := E.base.head.next; pos := 0;
  194.     WHILE u # E DO pos := pos + u.len; u := u.next END;
  195.     RETURN pos
  196. END ElemPos;
  197. PROCEDURE HandleAlien (E: Elem; VAR msg: ElemMsg);
  198.     VAR e: Alien; r: Files.Rider; i: LONGINT; ch: CHAR;
  199. BEGIN
  200.     WITH E: Alien DO
  201.         IF msg IS CopyMsg THEN
  202.             WITH msg: CopyMsg DO NEW(e); CopyElem(E, e);
  203.                 e.file := E.file; e.org := E.org; e.span := E.span; e.mod := E.mod; e.proc := E.proc;
  204.                 msg.e := e
  205.             END
  206.         ELSIF msg IS IdentifyMsg THEN
  207.             WITH msg: IdentifyMsg DO
  208.                 (*
  209.                     Is it a good idea to abuse the last element of the name
  210.                     to mark an alien ???
  211.                     Windows Oberon doesn't do that.
  212.                 *)
  213.                 COPY(E.mod, msg.mod); COPY(E.proc, msg.proc); msg.mod[31] := 1X (*alien*)
  214.             END
  215.         ELSIF msg IS FileMsg THEN
  216.             WITH msg: FileMsg DO
  217.                 IF msg.id = store THEN Files.Set(r, E.file, E.org); i := E.span;
  218.                     WHILE i > 0 DO Files.Read(r, ch); Files.Write(msg.r, ch); DEC(i) END
  219.                 END
  220.             END
  221.         END
  222. END HandleAlien;
  223. (** Buffers **)
  224. PROCEDURE OpenBuf* (B: Buffer);
  225.     VAR u: Run;
  226. BEGIN NEW(u); u.next := u; u.prev := u; B.head := u; B.len := 0
  227. END OpenBuf;
  228. PROCEDURE Copy* (SB, DB: Buffer);
  229.     VAR u, v, vn: Run;
  230. BEGIN u := SB.head.next; v := DB.head.prev;
  231.     WHILE u # SB.head DO
  232.         IF u IS Piece THEN vn := ClonePiece(u(Piece)) ELSE vn := CloneElem(u(Elem)) END;
  233.         v.next := vn; vn.prev := v; v := vn; u := u.next
  234.     END;
  235.     v.next := DB.head; DB.head.prev := v;
  236.     INC(DB.len, SB.len)
  237. END Copy;
  238. PROCEDURE Recall* (VAR B: Buffer);
  239. BEGIN B := del; del := NIL
  240. END Recall;
  241. (** Texts **)
  242. PROCEDURE Save* (T: Text; beg, end: LONGINT; B: Buffer);
  243.     VAR u, v, w, wn: Run; uo, ud, vo, vd: LONGINT;
  244. BEGIN Find(T, beg, u, uo, ud); Find(T, end, v, vo, vd);
  245.     w := B.head.prev;
  246.     WHILE u # v DO
  247.         IF u IS Piece THEN wn := ClonePiece(u(Piece)); DEC(wn.len, ud); INC(wn(Piece).org, ud)
  248.         ELSE wn := CloneElem(u(Elem))
  249.         END;
  250.         w.next := wn; wn.prev := w; w := wn; u := u.next; ud := 0
  251.     END;
  252.     IF vd > 0 THEN (*v IS Piece*) wn := ClonePiece(v(Piece)); wn.len := vd - ud; INC(wn(Piece).org, ud);
  253.         w.next := wn; wn.prev := w; w := wn
  254.     END;
  255.     w.next := B.head; B.head.prev := w;
  256.     INC(B.len, end - beg)
  257. END Save;
  258. PROCEDURE Insert* (T: Text; pos: LONGINT; B: Buffer);
  259.     VAR u, un, v: Run; uo, ud, len: LONGINT;
  260. BEGIN Find(T, pos, u, uo, ud); Split(ud, u, un);
  261.     len := B.len; v := B.head.next;
  262.     Merge(T, u, v); Splice(un, v, B.head.prev, T);
  263.     INC(T.len, len); B.head.next := B.head; B.head.prev := B.head; B.len := 0;
  264.     IF T.notify # NIL THEN T.notify(T, insert, pos, pos + len); END;
  265.     Display.Synchronize;
  266. END Insert;
  267. PROCEDURE Append* (T: Text; B: Buffer);
  268.     VAR v: Run; pos, len: LONGINT;
  269. BEGIN pos := T.len; len := B.len; v := B.head.next;
  270.     Merge(T, T.head.prev, v); Splice(T.head, v, B.head.prev, T);
  271.     INC(T.len, len); B.head.next := B.head; B.head.prev := B.head; B.len := 0;
  272.     IF T.notify # NIL THEN T.notify(T, insert, pos, pos + len); END;
  273.     Display.Synchronize;
  274. END Append;
  275. PROCEDURE Delete* (T: Text; beg, end: LONGINT);
  276.     VAR c, u, un, v, vn: Run; co, uo, ud, vo, vd: LONGINT;
  277. BEGIN
  278.     Find(T, beg, u, uo, ud); Split(ud, u, un); c := T.cache; co := T.corg;
  279.     Find(T, end, v, vo, vd); Split(vd, v, vn); T.cache := c; T.corg := co;
  280.     NEW(del); OpenBuf(del); del.len := end - beg;
  281.     Splice(del.head, un, v, NIL);
  282.     Merge(T, u, vn); u.next := vn; vn.prev := u;
  283.     DEC(T.len, end - beg);
  284.     IF T.notify # NIL THEN T.notify(T, delete, beg, end); END;
  285. END Delete;
  286. PROCEDURE ChangeLooks* (T: Text; beg, end: LONGINT; sel: SET; fnt: Fonts.Font; col, voff: SHORTINT);
  287.     VAR c, u, un, v, vn: Run; co, uo, ud, vo, vd: LONGINT;
  288. BEGIN Find(T, beg, u, uo, ud); Split(ud, u, un); c := T.cache; co := T.corg;
  289.     Find(T, end, v, vo, vd); Split(vd, v, vn); T.cache := c; T.corg := co;
  290.     WHILE un # vn DO
  291.         IF (0 IN sel) & (fnt # NIL) THEN un.fnt := fnt END;
  292.         IF 1 IN sel THEN un.col := col END;
  293.         IF 2 IN sel THEN un.voff := voff END;
  294.         Merge(T, u, un);
  295.         IF u.next = un THEN u := un; un := un.next ELSE u.next := un; un.prev := u END
  296.     END;
  297.     Merge(T, u, un); u.next := un; un.prev := u;
  298.     IF T.notify # NIL THEN T.notify(T, replace, beg, end); END;
  299. END ChangeLooks;
  300. (** Readers **)
  301. PROCEDURE OpenReader* (VAR R: Reader; T: Text; pos: LONGINT);
  302.     VAR u: Run;
  303. BEGIN
  304.     IF pos >= T.len THEN pos := T.len END;
  305.     Find(T, pos, u, R.org, R.off); R.run := u; R.eot := FALSE;
  306.     IF u IS Piece THEN
  307.         Files.Set(R.rider, u(Piece).file, u(Piece).org + R.off)
  308. END OpenReader;
  309. PROCEDURE Read* (VAR R: Reader; VAR ch: CHAR);
  310.     VAR u: Run;
  311. BEGIN u := R.run; R.fnt := u.fnt; R.col := u.col; R.voff := u.voff; INC(R.off);
  312.     IF u IS Piece THEN Files.Read(R.rider, ch); R.elem := NIL;
  313.     IF u(Piece).host THEN ch:=HostSYS.toOberon(ch); END;    (* << Unix: LF to CR *)
  314.     ELSIF u IS Elem THEN ch := ElemChar; R.elem := u(Elem)
  315.     ELSE ch := 0X; R.elem := NIL; R.eot := TRUE
  316.     END;
  317.     IF R.off = u.len THEN INC(R.org, u.len); u := u.next;
  318.         IF u IS Piece THEN
  319.             WITH u: Piece DO Files.Set(R.rider, u.file, u.org) END
  320.         END;
  321.         R.run := u; R.off := 0
  322. END Read;
  323. PROCEDURE ReadElem* (VAR R: Reader);
  324.     VAR u, un: Run;
  325. BEGIN u := R.run;
  326.     WHILE u IS Piece DO INC(R.org, u.len); u := u.next END;
  327.     IF u IS Elem THEN un := u.next; R.run := un; INC(R.org); R.off := 0;
  328.         R.fnt := u.fnt; R.col := u.col; R.voff := u.voff; R.elem := u(Elem);
  329.         IF un IS Piece THEN
  330.             WITH un: Piece DO Files.Set(R.rider, un.file, un.org) END
  331.         END
  332.     ELSE R.eot := TRUE; R.elem := NIL
  333. END ReadElem;
  334. PROCEDURE ReadPrevElem* (VAR R: Reader);
  335.     VAR u: Run;
  336. BEGIN u := R.run.prev;
  337.     WHILE u IS Piece DO DEC(R.org, u.len); u := u.prev END;
  338.     IF u IS Elem THEN R.run := u; DEC(R.org); R.off := 0;
  339.         R.fnt := u.fnt; R.col := u.col; R.voff := u.voff; R.elem := u(Elem)
  340.     ELSE R.eot := TRUE; R.elem := NIL
  341. END ReadPrevElem;
  342. PROCEDURE Pos* (VAR R: Reader): LONGINT;
  343. BEGIN RETURN R.org + R.off
  344. END Pos;
  345. (** Scanners --------------- NW --------------- **)
  346. PROCEDURE OpenScanner* (VAR S: Scanner; T: Text; pos: LONGINT);
  347. BEGIN OpenReader(S, T, pos); S.line := 0; S.nextCh := " "
  348. END OpenScanner;
  349.     IEEE floating-point formats  (BM 1992.1.1): (-1)^s * 1.m * 2^(e-e0), where
  350.     s    e    e0    m
  351. REAL    1-bit    8-bit biased    127    1+23-bit explicit
  352. LONGREAL    1-bit    11-bit biased    1023    1+52-bit explicit
  353. PROCEDURE Scan* (VAR S: Scanner);
  354.     CONST maxD = 256;
  355.     VAR ch: CHAR;
  356.         neg, negE, hex: BOOLEAN;
  357.         i, j, h: SHORTINT;
  358.         e: INTEGER; k: LONGINT;
  359.         x, f: REAL; y, g: LONGREAL;
  360.         d: ARRAY maxD OF CHAR;
  361.     PROCEDURE ReadScaleFactor;
  362.     BEGIN Read(S, ch);
  363.         IF ch = "-" THEN negE := TRUE; Read(S, ch)
  364.         ELSE negE := FALSE;
  365.             IF ch = "+" THEN Read(S, ch) END
  366.         END;
  367.         WHILE ("0" <= ch) & (ch <= "9") DO
  368.             e := e*10 + ORD(ch) - 30H; Read(S, ch)
  369.         END
  370.     END ReadScaleFactor;
  371. BEGIN ch := S.nextCh; i := 0;
  372.     LOOP
  373.         IF ch = CR THEN INC(S.line)
  374.         ELSIF (ch # " ") & (ch # TAB) THEN EXIT
  375.         END ;
  376.         Read(S, ch)
  377.     END;
  378.     IF HostSYS.IsNameChar1(ch) THEN (*name*)
  379.         REPEAT S.s[i] := ch; INC(i); Read(S, ch)
  380.         UNTIL ~HostSYS.IsNameChar(ch) OR (i = LEN(S.s)-1);
  381.         S.s[i] := 0X; S.len := i; S.class := Name;
  382.     ELSIF ch = 22X THEN (*literal string*)
  383.         Read(S, ch);
  384.         WHILE (ch # 22X) & (ch >= " ") & (i # LEN(S.s)-1) DO    (* << Unix *)
  385.             S.s[i] := ch; INC(i); Read(S, ch)
  386.         END;
  387.         S.s[i] := 0X; S.len := i; Read(S, ch); S.class := String;
  388.     ELSE
  389.         IF ch = "-" THEN neg := TRUE; Read(S, ch) ELSE neg := FALSE END ;
  390.         IF ("0" <= ch) & (ch <= "9") THEN (*number*)
  391.             hex := FALSE; j := 0;
  392.             LOOP d[i] := ch; INC(i); Read(S, ch);
  393.                 IF ch < "0" THEN EXIT END;
  394.                 IF "9" < ch THEN
  395.                     IF ("A" <= ch) & (ch <= "F") THEN hex := TRUE; ch := CHR(ORD(ch)-7)
  396.                     ELSIF ("a" <= ch) & (ch <= "f") THEN hex := TRUE; ch := CHR(ORD(ch)-27H)
  397.                     ELSE EXIT
  398.                     END
  399.                 END
  400.             END;
  401.             IF ch = "H" THEN (*hex number*)
  402.                 Read(S, ch); S.class := Int;
  403.                 IF i-j > 8 THEN j := i-8 END ;
  404.                 k := ORD(d[j]) - 30H; INC(j);
  405.                 IF (i-j = 7) & (k >= 8) THEN DEC(k, 16) END ;
  406.                 WHILE j < i DO k := k*10H + (ORD(d[j]) - 30H); INC(j) END ;
  407.                 IF neg THEN S.i := -k ELSE S.i := k END 
  408.             ELSIF ch = "." THEN (*read real*)
  409.                 Read(S, ch); h := i;
  410.                 WHILE ("0" <= ch) & (ch <= "9") DO d[i] := ch; INC(i); Read(S, ch) END ;
  411.                 IF ch = "D" THEN
  412.                     e := 0; y := 0; g := 1;
  413.                     REPEAT y := y*10 + (ORD(d[j]) - 30H); INC(j) UNTIL j = h;
  414.                     WHILE j < i DO g := g/10; y := (ORD(d[j]) - 30H)*g + y; INC(j) END ;
  415.                     ReadScaleFactor;
  416.                     IF negE THEN
  417.                         IF e <= 308 THEN y := y / Reals.TenL(e) ELSE y := 0 END
  418.                     ELSIF e > 0 THEN
  419.                         IF e <= 308 THEN y := Reals.TenL(e) * y ELSE HALT(40) END 
  420.                     END ;
  421.                     IF neg THEN y := -y END ;
  422.                     S.class := 5; S.y := y
  423.                 ELSE e := 0; x := 0; f := 1;
  424.                     REPEAT x := x*10 + (ORD(d[j]) - 30H); INC(j) UNTIL j = h;
  425.                     WHILE j < i DO f := f/10; x := (ORD(d[j])-30H)*f + x; INC(j) END;
  426.                     IF ch = "E" THEN ReadScaleFactor END ;
  427.                     IF negE THEN
  428.                         IF e <= 38 THEN x := x / Reals.Ten(e) ELSE x := 0 END
  429.                     ELSIF e > 0 THEN
  430.                         IF e <= 38 THEN x := Reals.Ten(e) * x ELSE HALT(40) END
  431.                     END ;
  432.                     IF neg THEN x := -x END ;
  433.                     S.class := LongReal; S.x := x
  434.                 END ;
  435.                 IF hex THEN S.class := Inval END
  436.             ELSE (*decimal integer*)
  437.                 S.class := Int; k := 0;
  438.                 REPEAT k := k*10 + (ORD(d[j]) - 30H); INC(j) UNTIL j = i;
  439.                 IF neg THEN S.i := -k ELSE S.i := k END;
  440.                 IF hex THEN S.class := 0 ELSE S.class := Int END
  441.             END
  442.         ELSE S.class := Char;
  443.             IF neg THEN S.c := "-" ELSE S.c := ch; Read(S, ch) END
  444.         END
  445.     END;
  446.     S.nextCh := ch
  447. END Scan;
  448. (** Writers **)
  449. PROCEDURE OpenWriter* (VAR W: Writer);
  450. BEGIN NEW(W.buf); OpenBuf(W.buf);
  451.     W.fnt := Fonts.Default; W.col := Display.white; W.voff := 0;
  452.     W.file := Files.New(""); Files.Set(W.rider, W.file, 0)
  453. END OpenWriter;
  454. PROCEDURE SetFont* (VAR W: Writer; fnt: Fonts.Font);
  455. BEGIN W.fnt := fnt
  456. END SetFont;
  457. PROCEDURE SetColor* (VAR W: Writer; col: SHORTINT);
  458. BEGIN W.col := col
  459. END SetColor;
  460. PROCEDURE SetOffset* (VAR W: Writer; voff: SHORTINT);
  461. BEGIN W.voff := voff
  462. END SetOffset;
  463. PROCEDURE Write* (VAR W: Writer; ch: CHAR);
  464.     VAR u, un: Run; p: Piece;
  465. BEGIN
  466.     Files.Write(W.rider, ch); INC(W.buf.len); un := W.buf.head; u := un.prev;
  467.     IF (u IS Piece) & (u(Piece).file = W.file) & (u.fnt.name = W.fnt.name) & (u.col = W.col) & (u.voff = W.voff)
  468.     & ~u(Piece).host THEN
  469.         INC(u.len)
  470.     ELSE NEW(p); u.next := p; p.prev := u; p.next := un; un.prev := p;
  471.         p.len := 1; p.fnt := W.fnt; p.col := W.col; p.voff := W.voff;
  472.         p.file := W.file; p.org := Files.Length(W.file) - 1; p.host := FALSE
  473. END Write;
  474. PROCEDURE WriteElem* (VAR W: Writer; e: Elem);
  475.     VAR u, un: Run;
  476. BEGIN
  477.     IF e.base # NIL THEN HALT(99) END;
  478.     INC(W.buf.len); e.len := 1; e.fnt := W.fnt; e.col := W.col; e.voff := W.voff;
  479.     un := W.buf.head; u := un.prev; u.next := e; e.prev := u; e.next := un; un.prev := e
  480. END WriteElem;
  481. PROCEDURE WriteLn* (VAR W: Writer);
  482. BEGIN Write(W, CR)
  483. END WriteLn;
  484. PROCEDURE WriteString* (VAR W: Writer; s: ARRAY OF CHAR);
  485.     VAR i: INTEGER;
  486. BEGIN i := 0;
  487.     WHILE s[i] >= " " DO Write(W, s[i]); INC(i) END
  488. END WriteString;
  489. PROCEDURE WriteInt* (VAR W: Writer; x, n: LONGINT);
  490.     VAR i: INTEGER; x0: LONGINT;
  491.         a: ARRAY 11 OF CHAR;
  492. BEGIN i := 0;
  493.     IF x < 0 THEN
  494.         IF x = MIN(LONGINT) THEN WriteString(W, " -2147483648"); RETURN
  495.         ELSE DEC(n); x0 := -x
  496.         END
  497.     ELSE x0 := x
  498.     END;
  499.     REPEAT
  500.         a[i] := CHR(x0 MOD 10 + 30H); x0 := x0 DIV 10; INC(i)
  501.     UNTIL x0 = 0;
  502.     WHILE n > i DO Write(W, " "); DEC(n) END;
  503.     IF x < 0 THEN Write(W, "-") END;
  504.     REPEAT DEC(i); Write(W, a[i]) UNTIL i = 0
  505. END WriteInt;
  506. PROCEDURE WriteHex* (VAR W: Writer; x: LONGINT);
  507.     VAR i: INTEGER; y: LONGINT;
  508.         a: ARRAY 10 OF CHAR;
  509. BEGIN i := 0; Write(W, " ");
  510.     REPEAT y := x MOD 10H;
  511.         IF y < 10 THEN a[i] := CHR(y + 30H) ELSE a[i] := CHR(y + 37H) END;
  512.         x := x DIV 10H; INC(i)
  513.     UNTIL i = 8;
  514.     REPEAT DEC(i); Write(W, a[i]) UNTIL i = 0
  515. END WriteHex;
  516. PROCEDURE WriteReal* (VAR W: Writer; x: REAL; n: INTEGER);
  517.     VAR e: INTEGER; x0: REAL;
  518.         d: ARRAY maxD OF CHAR;
  519. BEGIN e := Reals.Expo(x);
  520.     IF e = 0 THEN
  521.         WriteString(W, "  0");
  522.         REPEAT Write(W, " "); DEC(n) UNTIL n <= 3
  523.     ELSIF e = 255 THEN
  524.         WriteString(W, " NaN");
  525.         WHILE n > 4 DO Write(W, " "); DEC(n) END
  526.     ELSE
  527.         IF n <= 9 THEN n := 3 ELSE DEC(n, 6) END;
  528.         REPEAT Write(W, " "); DEC(n) UNTIL n <= 8;
  529.         (*there are 2 < n <= 8 digits to be written*)
  530.         IF x < 0.0 THEN Write(W, "-"); x := -x ELSE Write(W, " ") END;
  531.         e := (e - 127) * 77  DIV 256;
  532.         IF e >= 0 THEN x := x / Reals.Ten(e) ELSE x := Reals.Ten(-e) * x END;
  533.         IF x >= 10.0 THEN x := 0.1*x; INC(e) END;
  534.         x0 := Reals.Ten(n-1); x := x0*x + 0.5;
  535.         IF x >= 10.0*x0 THEN x := x*0.1; INC(e) END;
  536.         Reals.Convert(x, n, d);
  537.         DEC(n); Write(W, d[n]); Write(W, ".");
  538.         REPEAT DEC(n); Write(W, d[n]) UNTIL n = 0;
  539.         Write(W, "E");
  540.         IF e < 0 THEN Write(W, "-"); e := -e ELSE Write(W, "+") END;
  541.         Write(W, CHR(e DIV 10 + 30H)); Write(W, CHR(e MOD 10 + 30H))
  542. END WriteReal;
  543. PROCEDURE WriteRealFix* (VAR W: Writer; x: REAL; n, k: INTEGER);
  544.     VAR e, i: INTEGER; sign: CHAR; x0: REAL;
  545.         d: ARRAY maxD OF CHAR;
  546.     PROCEDURE seq(ch: CHAR; n: INTEGER);
  547.     BEGIN WHILE n > 0 DO Write(W, ch); DEC(n) END
  548.     END seq;
  549.     PROCEDURE dig(n: INTEGER);
  550.     BEGIN
  551.         WHILE n > 0 DO
  552.             DEC(i); Write(W, d[i]); DEC(n)
  553.         END
  554.     END dig;
  555. BEGIN e := Reals.Expo(x);
  556.     IF k < 0 THEN k := 0 END;
  557.     IF e = 0 THEN seq(" ", n-k-2); Write(W, "0"); seq(" ", k+1)
  558.     ELSIF e = 255 THEN WriteString(W, " NaN"); seq(" ", n-4)
  559.     ELSE e := (e - 127) * 77 DIV 256;
  560.         IF x < 0 THEN sign := "-"; x := -x ELSE sign := " " END;
  561.         IF e >= 0 THEN  (*x >= 1.0,  77/256 = log 2*) x := x/Reals.Ten(e)
  562.             ELSE (*x < 1.0*) x := Reals.Ten(-e) * x
  563.         END;
  564.         IF x >= 10.0 THEN x := 0.1*x; INC(e) END;
  565.         (* 1 <= x < 10 *)
  566.         IF k+e >= maxD-1 THEN k := maxD-1-e
  567.             ELSIF k+e < 0 THEN k := -e; x := 0.0
  568.         END;
  569.         x0 := Reals.Ten(k+e); x := x0*x + 0.5;
  570.         IF x >= 10.0*x0 THEN INC(e) END;
  571.         (*e = no. of digits before decimal point*)
  572.         INC(e); i := k+e; Reals.Convert(x, i, d);
  573.         IF e > 0 THEN
  574.             seq(" ", n-e-k-2); Write(W, sign); dig(e);
  575.             Write(W, "."); dig(k)
  576.         ELSE seq(" ", n-k-3);
  577.             Write(W, sign); Write(W, "0"); Write(W, ".");
  578.             seq("0", -e); dig(k+e)
  579.         END
  580. END WriteRealFix;
  581. PROCEDURE WriteRealHex* (VAR W: Writer; x: REAL);
  582.     VAR i: INTEGER;
  583.         d: ARRAY 8 OF CHAR;
  584. BEGIN Reals.ConvertH(x, d); i := 0;
  585.     REPEAT Write(W, d[i]); INC(i) UNTIL i = 8
  586. END WriteRealHex;
  587. PROCEDURE WriteLongReal* (VAR W: Writer; x: LONGREAL; n: INTEGER);
  588. CONST
  589.     maxD = 40;
  590.     e: INTEGER; x0: LONGREAL;
  591.     d: ARRAY maxD OF CHAR;
  592. BEGIN
  593.     e := Reals.ExpoL(x);
  594.     IF e = 0 THEN
  595.         WriteString(W, "  0");
  596.         REPEAT Write(W, " "); DEC(n) UNTIL n <= 3;
  597.     ELSIF e = 2047 THEN
  598.         WriteString(W, " NaN");
  599.         WHILE n > 4 DO Write(W, " "); DEC(n) END;
  600.     ELSE
  601.         IF n <= 10 THEN n := 3 ELSE DEC(n, 7) END;
  602.         REPEAT Write(W, " "); DEC(n) UNTIL n <= maxD;
  603.         (*there are 2 <= n <= maxD digits to be written*)
  604.         IF x < 0 THEN Write(W, "-"); x := -x ELSE Write(W, " ") END;
  605.         e := SHORT(LONG(e - 1023) * 77 DIV 256);
  606.         IF e >= 0 THEN x := x / Reals.TenL(e) ELSE x := Reals.TenL(-e) * x END ;
  607.         IF x >= 10.0D0 THEN x := 0.1D0 * x; INC(e) END ;
  608.         x0 := Reals.TenL(n-1); x := x0*x + 0.5D0;
  609.         IF x >= 10.0D0*x0 THEN x := 0.1D0 * x; INC(e) END ;
  610.         Reals.ConvertL(x, n, d);
  611.         DEC(n); Write(W, d[n]); Write(W, ".");
  612.         REPEAT DEC(n); Write(W, d[n]) UNTIL n = 0;
  613.         Write(W, "D");
  614.         IF e < 0 THEN Write(W, "-"); e := -e ELSE Write(W, "+") END;
  615.         Write(W, CHR(e DIV 100 + 30H)); e := e MOD 100;
  616.         Write(W, CHR(e DIV 10 + 30H));
  617.         Write(W, CHR(e MOD 10 + 30H))
  618. END WriteLongReal;
  619. PROCEDURE WriteLongRealHex* (VAR W: Writer; x: LONGREAL);
  620.     VAR i: INTEGER;
  621.         d: ARRAY 16 OF CHAR;
  622. BEGIN Reals.ConvertHL(x, d); i := 0;
  623.     REPEAT Write(W, d[i]); INC(i) UNTIL i = 16
  624. END WriteLongRealHex;
  625. PROCEDURE WriteDate* (VAR W: Writer; t, d: LONGINT);
  626.     PROCEDURE WritePair(ch: CHAR; x: LONGINT);
  627.     BEGIN Write(W, ch);
  628.       Write(W, CHR(x DIV 10 + 30H)); Write(W, CHR(x MOD 10 + 30H))
  629.     END WritePair;
  630. BEGIN
  631.     WritePair(" ", d MOD 32); WritePair(".", d DIV 32 MOD 16); WritePair(".", d DIV 512 MOD 128);
  632.     WritePair(" ", t DIV 4096 MOD 32); WritePair(":", t DIV 64 MOD 64); WritePair(":", t MOD 64)
  633. END WriteDate;
  634. (** Text Filing **)
  635. PROCEDURE Load0 (VAR r: Files.Rider; T: Text);
  636.     VAR u, un: Run; p: Piece; e: Elem;
  637.         org, pos, hlen, plen: LONGINT; ecnt, fno, fcnt, col, voff: SHORTINT;
  638.         f: Files.File;
  639.         msg: FileMsg;
  640.         mods, procs: ARRAY 64, 32 OF CHAR;
  641.         name: ARRAY 32 OF CHAR;
  642.         fnts: ARRAY 32 OF Fonts.Font;
  643.     PROCEDURE LoadElem (VAR r: Files.Rider; pos, span: LONGINT; VAR e: Elem);
  644.         VAR M: Modules.Module; Cmd: Modules.Command; a: Alien;
  645.             org, ew, eh: LONGINT; eno: SHORTINT;
  646.     BEGIN new := NIL;
  647.         Files.ReadLInt(r, ew); Files.ReadLInt(r, eh); Files.Read(r, eno);
  648.         IF eno > ecnt THEN ecnt := eno; Files.ReadString(r, mods[eno]); Files.ReadString(r, procs[eno]) END;
  649.         org := Files.Pos(r); M := Modules.ThisMod(mods[eno]);
  650.         IF M # NIL THEN Cmd := Modules.ThisCommand(M, procs[eno]);
  651.             IF Cmd # NIL THEN Cmd END
  652.         END;
  653.         e := new;
  654.         IF e # NIL THEN e.W := ew; e.H := eh; e.base := T;
  655.             msg.pos := pos; e.handle(e, msg);
  656.             IF Files.Pos(r) # org + span THEN e := NIL END
  657.         END;
  658.         IF e = NIL THEN Files.Set(r, f, org + span);
  659.             NEW(a); a.W := ew; a.H := eh; a.handle := HandleAlien; a.base := T;
  660.             a.file := f; a.org := org; a.span := span;
  661.             COPY(mods[eno], a.mod); COPY(procs[eno], a.proc);
  662.             e := a
  663.         END
  664.     END LoadElem;
  665. BEGIN pos := Files.Pos(r); f := Files.Base(r);
  666.     NEW(u); u.len := MAX(LONGINT); (*u.fnt := Fonts.Default;*)u.fnt := NIL; u.col := Display.white;
  667.     T.head := u; ecnt := 0; fcnt := 0;
  668.     msg.id := load; msg.r := r;
  669.     Files.ReadLInt(msg.r, hlen); (*!!!org := pos + hlen;*) org := pos -2 + hlen; pos := org; Files.Read(msg.r, fno);
  670.     WHILE fno # 0 DO
  671.         IF fno > fcnt THEN fcnt := fno; Files.ReadString(msg.r, name); fnts[fno] := Fonts.This(name) END;
  672.         Files.Read(msg.r, col); Files.Read(msg.r, voff); Files.ReadLInt(msg.r, plen);
  673.         IF plen > 0 THEN NEW(p); p.file := f; p.org := pos; p.host := FALSE; un := p; un.len := plen
  674.         ELSE LoadElem(msg.r, pos - org, -plen, e); un := e; un.len := 1
  675.         END;
  676.         un.fnt := fnts[fno]; un.col := col; un.voff := voff;
  677.         INC(pos, un.len); u.next := un; un.prev := u; u := un; Files.Read(msg.r, fno)
  678.     END;
  679.     u.next := T.head; T.head.prev := u; T.cache := T.head; T.corg := 0;
  680.     Files.ReadLInt(msg.r, T.len); Files.Set(r, f, Files.Pos(msg.r) + T.len)
  681. END Load0;
  682. PROCEDURE Load* (VAR r: Files.Rider; T: Text);
  683.     CONST oldTag = -4095;
  684.     VAR tag: INTEGER;
  685. BEGIN
  686.     (* for compatibility inner text tags are checked and skipped; remove this in a later version *)
  687.     Files.ReadInt(r, tag); IF tag # oldTag THEN Files.Set(r, Files.Base(r), Files.Pos(r)-2) END;
  688.     Load0(r, T)
  689. END Load;
  690. PROCEDURE Open* (T: Text; name: ARRAY OF CHAR);
  691.     fixedFont:Fonts.Font;
  692.     f:Files.File;
  693.     r:Files.Rider; 
  694.     u:Run; 
  695.     p:Piece; 
  696.     tag,version: CHAR;
  697. BEGIN
  698.     f := Files.Old(name);
  699.     IF f = NIL THEN                        (* 17.2.95 mah do not read (buffer allocated) when opening new file *)
  700.         f := Files.New(""); tag := 0X
  701.     ELSE
  702.         Files.Set(r, f, 0); Files.Read(r, tag); Files.Read(r, version)
  703.     END;
  704.     IF (tag=textTag) & (version=textVersion) THEN
  705.         Load0(r,T);
  706.     ELSE
  707.             Not an Oberon Text file.
  708.         fixedFont:=Fonts.This("Courier10.Scn.Fnt");
  709.         NEW(u); u.len := MAX(LONGINT); u.fnt := NIL; u.col := Display.white;
  710.         T.len := Files.Length(f);
  711.         IF T.len > 0 THEN
  712.             NEW(p); p.len := T.len; p.fnt := fixedFont;
  713.             p.col := Display.white; p.voff := 0; p.file := f; p.org := 0; p.host := TRUE;
  714.             u.next := p; u.prev := p; p.next := u; p.prev := u
  715.         ELSE
  716.             u.next := u; u.prev := u
  717.         END;
  718.         T.head := u; T.cache := T.head; T.corg := 0
  719. END Open;
  720. PROCEDURE Store* (VAR r: Files.Rider; T: Text);
  721.     VAR r1: Files.Rider; u, un: Run; org, pos, delta, hlen, rlen: LONGINT; ecnt, fno, fcnt: SHORTINT;
  722.         ch: CHAR;
  723.         msg: FileMsg; iden: IdentifyMsg;
  724.         mods, procs: ARRAY 64, 32 OF CHAR;
  725.         fnts: ARRAY 32 OF Fonts.Font;
  726.         block: ARRAY 1024 OF CHAR;
  727.     PROCEDURE StoreElem (VAR r: Files.Rider; pos: LONGINT; e: Elem);
  728.         Windows StoreElem stores elem Width and Height before sending the
  729.         FileMsg. This makes a difference, if a handler changes W and H when
  730.         it receives the FileMsg. Is this allowed?
  731.         VAR r1: Files.Rider; org, span: LONGINT; eno: SHORTINT;
  732.     BEGIN COPY(iden.mod, mods[ecnt]); COPY(iden.proc, procs[ecnt]); eno := 1;
  733.         WHILE (mods[eno] # iden.mod) OR (procs[eno] # iden.proc) DO INC(eno) END;
  734.         Files.Set(r1, Files.Base(r), Files.Pos(r));
  735.         Files.WriteLInt(r, 0); Files.WriteLInt(r, 0); Files.WriteLInt(r, 0); (*fixup slot*)
  736.         Files.Write(r, eno);
  737.         IF eno = ecnt THEN INC(ecnt); Files.WriteString(r, iden.mod); Files.WriteString(r, iden.proc) END;
  738.         msg.pos := pos; org := Files.Pos(r); e.handle(e, msg); span := Files.Pos(r) - org;
  739.         Files.WriteLInt(r1, -span); Files.WriteLInt(r1, e.W); Files.WriteLInt(r1, e.H) (*fixup*)
  740.     END StoreElem;
  741. BEGIN
  742.     org := Files.Pos(r); msg.id := store; msg.r := r; Files.WriteLInt(msg.r, 0); (*fixup slot*)
  743.     u := T.head.next; pos := 0; delta := 0; fcnt := 1; ecnt := 1;
  744.     WHILE u # T.head DO
  745.         IF u IS Elem THEN iden.mod[0] := 0X; u(Elem).handle(u(Elem), iden) ELSE iden.mod[0] := 1X END;
  746.         IF iden.mod[0] # 0X THEN
  747.             fnts[fcnt] := u.fnt; fno := 1;
  748.             WHILE fnts[fno].name # u.fnt.name DO INC(fno) END;
  749.             Files.Write(msg.r, fno);
  750.             IF fno = fcnt THEN INC(fcnt); Files.WriteString(msg.r, u.fnt.name) END;
  751.             Files.Write(msg.r, u.col); Files.Write(msg.r, u.voff)
  752.         END;
  753.         IF u IS Piece THEN rlen := u.len; un := u.next;
  754.             WHILE (un IS Piece) & (un.fnt = u.fnt) & (un.col = u.col) & (un.voff = u.voff) DO
  755.                 INC(rlen, un.len); un := un.next
  756.             END;
  757.             Files.WriteLInt(msg.r, rlen); INC(pos, rlen); u := un
  758.         ELSIF iden.mod[0] # 0X THEN StoreElem(msg.r, pos, u(Elem)); INC(pos); u := u.next
  759.         ELSE INC(delta); u := u.next
  760.         END
  761.     END;
  762.     Files.Write(msg.r, 0); Files.WriteLInt(msg.r, T.len - delta);
  763.     (*!!!hlen := Files.Pos(msg.r) - org;*) hlen := Files.Pos(msg.r) - org + 2;
  764.     Files.Set(r1, Files.Base(msg.r), org); Files.WriteLInt(r1, hlen); (*fixup*)
  765.     u := T.head.next;
  766.     WHILE u # T.head DO
  767.         IF u IS Piece THEN
  768.             WITH u: Piece DO
  769.                 IF u.host THEN Files.Set(r1, u.file, u.org); delta := u.len;
  770.                     WHILE delta > 0 DO Files.Read(r1, ch); DEC(delta);
  771.                         Files.Write(msg.r, HostSYS.toOberon(ch));
  772.                     END
  773.                 ELSE Files.Set(r1, u.file, u.org); delta := u.len;
  774.                     WHILE delta > LEN(block) DO Files.ReadBytes(r1, block, LEN(block));
  775.                         Files.WriteBytes(msg.r, block, LEN(block)); DEC(delta, LEN(block))
  776.                     END;
  777.                     Files.ReadBytes(r1, block, delta); Files.WriteBytes(msg.r, block, delta)
  778.                 END
  779.             END
  780.         ELSE iden.mod[0] := 0X; u(Elem).handle(u(Elem), iden);
  781.             IF iden.mod[0] # 0X THEN Files.Write(msg.r, ElemChar) END
  782.         END;
  783.         u := u.next
  784.     END;
  785.     r := msg.r;
  786. END Store;
  787. PROCEDURE Close* (T: Text; name: ARRAY OF CHAR);
  788.     VAR f: Files.File; r: Files.Rider; i, res: INTEGER; bak: ARRAY 512 OF CHAR;
  789. BEGIN
  790.     f := Files.New(name); Files.Set(r, f, 0); Files.Write(r, textTag); Files.Write(r,textVersion); Store(r, T);
  791.     i := 0; WHILE name[i] # 0X DO INC(i) END;
  792.     COPY(name, bak); bak[i] := "."; bak[i+1] := "B"; bak[i+2] := "a"; bak[i+3] := "k"; bak[i+4] := 0X;
  793.     Files.Rename(name, bak, res); Files.Register(f)
  794. END Close;
  795. BEGIN
  796.     del:=NIL;
  797.     isNameChar1:=HostSYS.IsNameChar1;
  798.     isNameChar:=HostSYS.IsNameChar;
  799. END Texts.
  800. Date    Author    Modification
  801. 1996-07-31    claudio@dial.eunet.ch    First unified version.
  802.